home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / zpacked.c < prev    next >
C/C++ Source or Header  |  1995-01-24  |  6KB  |  233 lines

  1. /* Copyright (C) 1990, 1992, 1993 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* zpacked.c */
  20. /* Packed array operators */
  21. #include "ghost.h"
  22. #include "errors.h"
  23. #include "ialloc.h"
  24. #include "idict.h"
  25. #include "iname.h"
  26. #include "istack.h"        /* for iparray.h */
  27. #include "ipacked.h"
  28. #include "iparray.h"
  29. #include "ivmspace.h"
  30. #include "oper.h"
  31. #include "store.h"
  32.  
  33. /* Import the array packing flag */
  34. extern ref ref_array_packing;
  35.  
  36. /* - currentpacking <bool> */
  37. private int
  38. zcurrentpacking(register os_ptr op)
  39. {    push(1);
  40.     ref_assign(op, &ref_array_packing);
  41.     return 0;
  42. }
  43.  
  44. /* <obj_0> ... <obj_n-1> <n> packedarray <packedarray> */
  45. int
  46. zpackedarray(register os_ptr op)
  47. {    int code;
  48.     ref parr;
  49.     check_type(*op, t_integer);
  50.     if ( op->value.intval < 0 ||
  51.          (op->value.intval > op - osbot &&
  52.           op->value.intval >= ref_stack_count(&o_stack))
  53.        )
  54.         return_error(e_rangecheck);
  55.     osp--;
  56.     code = make_packed_array(&parr, &o_stack, (uint)op->value.intval,
  57.                  "packedarray");
  58.     osp++;
  59.     if ( code >= 0 )
  60.       *osp = parr;
  61.     return code;
  62. }
  63.  
  64. /* <bool> setpacking - */
  65. private int
  66. zsetpacking(register os_ptr op)
  67. {    check_type(*op, t_boolean);
  68.     ref_assign_old(NULL, &ref_array_packing, op, "setpacking");
  69.     pop(1);
  70.     return 0;
  71. }
  72.  
  73. /* ------ Non-operator routines ------ */
  74.  
  75. /* Make a packed array.  See the comment in packed.h about */
  76. /* ensuring that refs in mixed arrays are properly aligned. */
  77. int
  78. make_packed_array(ref *parr, ref_stack *pstack, uint size, client_name_t cname)
  79. {    uint i;
  80.     const ref *pref;
  81.     uint idest = 0, ishort = 0;
  82.     ref_packed *pbody, *pdest;
  83.     ref_packed *pshort;        /* points to start of */
  84.                     /* last run of short elements */
  85.     uint space = ialloc_space(idmemory);
  86.     int skip = 0, pad;
  87.     ref rtemp;
  88.     int code;
  89.  
  90.     /* Do a first pass to calculate the size of the array, */
  91.     /* and to detect local-into-global stores. */
  92.  
  93.     for ( i = size; i != 0; i-- )
  94.     {    pref = ref_stack_index(pstack, i - 1);
  95.         switch ( r_btype(pref) )  /* not r_type, opers are special */
  96.         {
  97.         case t_name:
  98.             if ( name_index(pref) >= packed_name_max_index )
  99.                 break;    /* can't pack */
  100.             idest++;
  101.             continue;
  102.         case t_integer:
  103.             if ( pref->value.intval < packed_min_intval ||
  104.                  pref->value.intval > packed_max_intval
  105.                )
  106.                 break;
  107.             idest++;
  108.             continue;
  109.         case t_oparray:
  110.             /* Check for local-into-global store. */
  111.             store_check_space(space, pref);
  112.             /* falls through */
  113.         case t_operator:
  114.         {    uint oidx;
  115.             if ( !r_has_attr(pref, a_executable) )
  116.                 break;
  117.             oidx = op_index(pref);
  118.             if ( oidx == 0 || oidx > packed_int_mask )
  119.                 break;
  120.         }    idest++;
  121.             continue;
  122.         default:
  123.             /* Check for local-into-global store. */
  124.             store_check_space(space, pref);
  125.         }
  126.         /* Can't pack this element, use a full ref. */
  127.         /* We may have to unpack up to align_packed_per_ref - 1 */
  128.         /* preceding short elements. */
  129.         /* If we are at the beginning of the array, however, */
  130.         /* we can just move the elements up. */
  131.         {    int i = (idest - ishort) & (align_packed_per_ref - 1);
  132.             if ( ishort == 0 )        /* first time */
  133.               idest += skip = -i & (align_packed_per_ref - 1);
  134.             else
  135.               idest += (packed_per_ref - 1) * i;
  136.         }
  137.         ishort = idest += packed_per_ref;
  138.     }
  139.     pad = -idest & (packed_per_ref - 1);    /* padding at end */
  140.  
  141.     /* Now we can allocate the array. */
  142.  
  143.     code = ialloc_ref_array(&rtemp, 0, (idest + pad) / packed_per_ref,
  144.                 cname);
  145.     if ( code < 0 )
  146.       return code;
  147.     pbody = (ref_packed *)rtemp.value.refs;
  148.  
  149.     /* Make sure any initial skipped elements contain legal packed */
  150.     /* refs, so that the garbage collector can scan storage. */
  151.  
  152.     pshort = pbody;
  153.     for ( ; skip; skip-- )
  154.         *pbody++ = pt_tag(pt_integer);
  155.     pdest = pbody;
  156.  
  157.     for ( i = size; i != 0; i-- )
  158.     {    pref = ref_stack_index(pstack, i - 1);
  159.         switch ( r_btype(pref) )    /* not r_type, opers are special */
  160.         {
  161.         case t_name:
  162.         {    uint nidx = name_index(pref);
  163.             if ( nidx >= packed_name_max_index )
  164.                 break;    /* can't pack */
  165.             *pdest++ = nidx +
  166.               (r_has_attr(pref, a_executable) ?
  167.                pt_tag(pt_executable_name) :
  168.                pt_tag(pt_literal_name));
  169.         }    continue;
  170.         case t_integer:
  171.             if ( pref->value.intval < packed_min_intval ||
  172.                  pref->value.intval > packed_max_intval
  173.                )
  174.                 break;
  175.             *pdest++ = pt_tag(pt_integer) +
  176.               ((short)pref->value.intval - packed_min_intval);
  177.             continue;
  178.         case t_oparray:
  179.         case t_operator:
  180.         {    uint oidx;
  181.             if ( !r_has_attr(pref, a_executable) )
  182.                 break;
  183.             oidx = op_index(pref);
  184.             if ( oidx == 0 || oidx > packed_int_mask )
  185.                 break;
  186.             *pdest++ = pt_tag(pt_executable_operator) + oidx;
  187.         }    continue;
  188.         }
  189.         /* Can't pack this element, use a full ref. */
  190.         /* We may have to unpack up to align_packed_per_ref - 1 */
  191.         /* preceding short elements. */
  192.         /* Note that if we are at the beginning of the array, */
  193.         /* 'skip' already ensures that we don't need to do this. */
  194.         {    int i = (pdest - pshort) & (align_packed_per_ref - 1);
  195.             const ref_packed *psrc = pdest;
  196.             ref *pmove =
  197.               (ref *)(pdest += (packed_per_ref - 1) * i);
  198.             ref_assign_new(pmove, pref);
  199.             while ( --i >= 0 )
  200.             {    --psrc;
  201.                 --pmove;
  202.                 packed_get(psrc, pmove);
  203.             }
  204.         }
  205.         pshort = pdest += packed_per_ref;
  206.     }
  207.  
  208.     {    int atype =
  209.           (pdest == pbody + size ? t_shortarray : t_mixedarray);
  210.  
  211.         /* Pad with legal packed refs so that the garbage collector */
  212.         /* can scan storage. */
  213.  
  214.         for ( ; pad; pad-- )
  215.             *pdest++ = pt_tag(pt_integer);
  216.  
  217.         /* Finally, make the array. */
  218.  
  219.         ref_stack_pop(pstack, size);
  220.         make_tasv_new(parr, atype, a_readonly | space, size,
  221.                   packed, pbody + skip);
  222.     }
  223.     return 0;
  224. }
  225.  
  226. /* ------ Initialization procedure ------ */
  227.  
  228. BEGIN_OP_DEFS(zpacked_op_defs) {
  229.     {"0currentpacking", zcurrentpacking},
  230.     {"1packedarray", zpackedarray},
  231.     {"1setpacking", zsetpacking},
  232. END_OP_DEFS(0) }
  233.